home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Just Call Me Internet
/
Just Call Me Internet.iso
/
prog
/
atari
/
c
/
nos042_s
/
iface.c
< prev
next >
Wrap
C/C++ Source or Header
|
1994-09-16
|
13KB
|
557 lines
/*
IP interface control and configuration routines
Copyright 1991 Phil Karn, KA9Q
*/
/****************************************************************************
* $Id: iface.c 1.3 93/07/16 11:45:31 ROOT_DOS Exp $
* 04 Jul 93 1.2 GT Fix warnings. *
Atari version by David Nash - dnash@chaos.demon.co.uk
11.06.94 - add ifconfig up/down
****************************************************************************/
#include <stdio.h>
#include <ctype.h>
#include "config.h"
#include "pc.h"
#include "global.h"
#include "mbuf.h"
#include "proc.h"
#include "iface.h"
#include "ip.h"
#include "netuser.h"
#include "ax25.h"
#include "enet.h"
#include "pktdrvr.h"
#include "cmdparse.h"
#include "commands.h"
#include "devparam.h"
static void showiface __ARGS((struct iface *ifp));
static int mask2width __ARGS((int32 mask));
static int ifipaddr __ARGS((int argc,char *argv[],void *p));
static int iflinkadr __ARGS((int argc,char *argv[],void *p));
static int ifbroad __ARGS((int argc,char *argv[],void *p));
static int ifnetmsk __ARGS((int argc,char *argv[],void *p));
static int ifrxbuf __ARGS((int argc,char *argv[],void *p));
static int ifmtu __ARGS((int argc,char *argv[],void *p));
static int ifforw __ARGS((int argc,char *argv[],void *p));
static int ifencap __ARGS((int argc,char *argv[],void *p));
/* Interface list header */
struct iface *Ifaces = &Loopback;
/* Loopback pseudo-interface */
struct iface Loopback = {
&Encap, /* Link to next entry */
"loopback", /* name */
0x7f000001L, /* addr 127.0.0.1 */
0xffffffffL, /* broadcast 255.255.255.255 */
0xffffffffL, /* netmask 255.255.255.255 */
MAXINT16, /* mtu No limit */
0, /* flags */
0, /* trace */
NULLCHAR, /* trfile */
NULLFILE, /* trfp */
NULLIF, /* forw */
NULLPROC, /* rxproc */
NULLPROC, /* txproc */
NULLPROC, /* supv */
#ifdef FILTER
NULL, /* infilter */
NULL, /* outfilter */
#endif
0, /* dev */
NULL, /* (*ioctl) */
NULLFP, /* (*iostatus) */
NULLFP, /* (*stop) */
NULLCHAR, /* hwaddr */
NULL, /* extension */
CL_NONE, /* type */
0, /* xdev */
&Iftypes[0], /* iftype */
NULLFP, /* (*send) */
NULLFP, /* (*output) */
NULLFP, /* (*raw) */
NULLVFP, /* (*status) */
NULLFP, /* (*discard) */
NULLFP, /* (*echo) */
0, /* ipsndcnt */
0, /* rawsndcnt */
0, /* iprecvcnt */
0, /* rawrcvcnt */
0, /* lastsent */
0, /* lastrecv */
};
/* Encapsulation pseudo-interface */
struct iface Encap = {
NULLIF,
"encap", /* name */
INADDR_ANY, /* addr 0.0.0.0 */
0xffffffffL, /* broadcast 255.255.255.255 */
0xffffffffL, /* netmask 255.255.255.255 */
MAXINT16, /* mtu No limit */
0, /* flags */
0, /* trace */
NULLCHAR, /* trfile */
NULLFILE, /* trfp */
NULLIF, /* forw */
NULLPROC, /* rxproc */
NULLPROC, /* txproc */
NULLPROC, /* supv */
#ifdef FILTER
NULL, /* infilter */
NULL, /* outfilter */
#endif
0, /* dev */
NULL, /* (*ioctl) */
NULLFP, /* (*iostatus) */
NULLFP, /* (*stop) */
NULLCHAR, /* hwaddr */
NULL, /* extension */
CL_NONE, /* type */
0, /* xdev */
&Iftypes[0], /* iftype */
ip_encap, /* (*send) */
NULLFP, /* (*output) */
NULLFP, /* (*raw) */
NULLVFP, /* (*status) */
NULLFP, /* (*discard) */
NULLFP, /* (*echo) */
0, /* ipsndcnt */
0, /* rawsndcnt */
0, /* iprecvcnt */
0, /* rawrcvcnt */
0, /* lastsent */
0, /* lastrecv */
};
char Noipaddr[] = "IP address field missing, and ip address not set\n";
struct cmds Ifcmds[] = {
{ "broadcast", ifbroad, 0, 2, NULLCHAR },
{ "encapsulation", ifencap, 0, 2, NULLCHAR },
{ "forward", ifforw, 0, 2, NULLCHAR },
{ "ipaddress", ifipaddr, 0, 2, NULLCHAR },
{ "linkaddress", iflinkadr, 0, 2, NULLCHAR },
{ "mtu", ifmtu, 0, 2, NULLCHAR },
{ "netmask", ifnetmsk, 0, 2, NULLCHAR },
{ "rxbuf", ifrxbuf, 0, 2, NULLCHAR },
{ NULLCHAR },
};
/*
doifconfig - Set interface parameters
*/
int doifconfig(int argc, char *argv[], void *p)
{
int i;
struct iface *ifp;
if (argc < 2) {
for (ifp = Ifaces; ifp != NULLIF; ifp = ifp->next)
showiface(ifp);
return 0;
}
if ((ifp = if_lookup(argv[1])) == NULLIF) {
tprintf("Interface %s unknown\n", argv[1]);
return 1;
}
if (argc == 2) {
showiface(ifp);
if (ifp->show != 0 ) {
(*ifp->show)(ifp);
}
return 0;
}
if (argc == 3) {
if (tolower(*argv[2]) == 'u' && ifp->iostatus != 0) {
(*ifp->iostatus)(ifp, PARAM_UP, 0);
return 0;
}
if (tolower(*argv[2]) == 'd' && ifp->iostatus != 0) {
(*ifp->iostatus)(ifp, PARAM_DOWN, 0);
return 0;
}
tprintf("Argument missing\n");
return 1;
}
for (i = 2; i < argc-1; i += 2)
subcmd(Ifcmds, 3, &argv[i-1], ifp);
return 0;
}
/*
ifipaddr - Set interface IP address
*/
static int ifipaddr(int argc, char *argv[], void *p)
{
struct iface *ifp = p;
ifp->addr = resolve(argv[1]);
return 0;
}
/*
iflinkadr - Set link (hardware) address
*/
static int iflinkadr(int argc, char *argv[], void *p)
{
struct iface *ifp = p;
if (ifp->iftype == NULLIFT || ifp->iftype->scan == NULL) {
tprintf("Can't set link address\n");
return 1;
}
if (ifp->hwaddr != NULLCHAR)
free(ifp->hwaddr);
ifp->hwaddr = mallocw(ifp->iftype->hwalen);
(*ifp->iftype->scan)(ifp->hwaddr, argv[1]);
return 0;
}
/*
ifbroad - Set interface broadcast address. This is actually done
by installing a private entry in the routing table.
*/
static int ifbroad(int argc, char *argv[], void *p)
{
struct iface *ifp = p;
struct route *rp;
rp = rt_blookup(ifp->broadcast,32);
if (rp != NULLROUTE && rp->iface == ifp)
rt_drop(ifp->broadcast,32);
ifp->broadcast = resolve(argv[1]);
rt_add(ifp->broadcast, 32, 0L, ifp, 1L, 0L, 1);
return 0;
}
/*
ifnetmsj - Set the network mask. This is actually done by installing
a routing entry.
*/
static int ifnetmsk(int argc, char *argv[], void *p)
{
struct iface *ifp = p;
struct route *rp;
/* Remove old entry if it exists */
rp = rt_blookup(ifp->addr & ifp->netmask, mask2width(ifp->netmask));
if(rp != NULLROUTE)
rt_drop(rp->target, rp->bits);
ifp->netmask = htol(argv[1]);
rt_add(ifp->addr, mask2width(ifp->netmask), 0L, ifp, 0L, 0L, 0);
return 0;
}
/*
ifencap - Command to set interface encapsulation mode
*/
static int ifencap(int argc, char *argv[], void *p)
{
struct iface *ifp = p;
if (setencap(ifp, argv[1]) != 0){
tprintf("Encapsulation mode '%s' unknown\n", argv[1]);
return 1;
}
return 0;
}
/*
setencap - Function to set encapsulation mode
*/
int setencap(struct iface *ifp, char *mode)
{
struct iftype *ift;
for(ift = &Iftypes[0]; ift->name != NULLCHAR; ift++)
if (strnicmp(ift->name, mode, strlen(mode)) == 0)
break;
if (ift->name == NULLCHAR){
return -1;
}
ifp->iftype = ift;
ifp->send = ift->send;
ifp->output = ift->output;
ifp->type = ift->type;
return 0;
}
/* Set interface receive buffer size */
static int ifrxbuf(int argc, char *argv[], void *p)
{
return 0; /* To be written */
}
/*
ifmtu - Set interface Maximum Transmission Unit
*/
static int ifmtu(int argc, char *argv[], void *p)
{
struct iface *ifp = p;
ifp->mtu = atoi(argv[1]);
return 0;
}
/*
ifforw - Set interface forwarding
*/
static int ifforw(int argc, char *argv[], void *p)
{
struct iface *ifp = p;
ifp->forw = if_lookup(argv[1]);
if (ifp->forw == ifp)
ifp->forw = NULLIF;
return 0;
}
/*
showiface - Display the parameters for a specified interface */
static void showiface(struct iface *ifp)
{
char tmp[25];
tprintf("%-10s IP addr %s MTU %u Link encap ", ifp->name,
inet_ntoa(ifp->addr),(int)ifp->mtu);
if (ifp->iftype == NULLIFT) {
tprintf("not set\n");
} else {
tprintf("%s\n", ifp->iftype->name);
if (ifp->iftype->format != NULL && ifp->hwaddr != NULLCHAR)
tprintf(" Link addr %s\n",
(*ifp->iftype->format)(tmp,ifp->hwaddr));
}
tprintf(" flags %u trace 0x%x netmask 0x%08lx broadcast %s\n",
ifp->flags,ifp->trace,ifp->netmask,inet_ntoa(ifp->broadcast));
if (ifp->forw != NULLIF)
tprintf(" output forward to %s\n", ifp->forw->name);
tprintf(" sent: ip %lu tot %lu idle %s\n",
ifp->ipsndcnt, ifp->rawsndcnt, tformat(secclock() - ifp->lastsent));
tprintf(" recv: ip %lu tot %lu idle %s\n",
ifp->iprecvcnt, ifp->rawrecvcnt, tformat(secclock() - ifp->lastrecv));
}
/*
dodetach - Command to detach an interface
*/
int dodetach(int argc, char *argv[], void *p)
{
struct iface *ifp;
if ((ifp = if_lookup(argv[1])) == NULLIF) {
tprintf("Interface %s unknown\n", argv[1]);
return 1;
}
if (if_detach(ifp) == -1)
tprintf("Can't detach loopback or encap interface\n");
return 0;
}
/*
if_detach - Detach a specified interface
*/
int if_detach(struct iface *ifp)
{
int i, j;
struct iface *iftmp;
struct route *rp,*rptmp;
if (ifp == &Loopback || ifp == &Encap)
return -1;
/* Drop all routes that point to this interface */
if (R_default.iface == ifp)
rt_drop(0L, 0); /* Drop default route */
for (i=0; i < HASHMOD; i++) {
for (j = 0; j < 32; j++) {
for (rp = Routes[j][i]; rp != NULLROUTE; rp = rptmp) {
/* Save next pointer in case we delete this entry */
rptmp = rp->next;
if (rp->iface == ifp)
rt_drop(rp->target, rp->bits);
}
}
}
/* Unforward any other interfaces forwarding to this one */
for (iftmp = Ifaces; iftmp != NULLIF; iftmp = iftmp->next) {
if (iftmp->forw == ifp)
iftmp->forw = NULLIF;
}
/* Call device shutdown routine, if any */
if (ifp->stop != NULLFP)
(*ifp->stop)(ifp);
killproc(ifp->rxproc);
killproc(ifp->txproc);
killproc(ifp->supv);
/* Free allocated memory associated with this interface */
if (ifp->name != NULLCHAR)
free(ifp->name);
if (ifp->hwaddr != NULLCHAR)
free(ifp->hwaddr);
/* Remove from interface list */
if (ifp == Ifaces){
Ifaces = ifp->next;
} else {
/* Search for entry just before this one
* (necessary because list is only singly-linked.)
*/
for (iftmp = Ifaces; iftmp != NULLIF; iftmp = iftmp->next)
if(iftmp->next == ifp)
break;
if (iftmp != NULLIF && iftmp->next == ifp)
iftmp->next = ifp->next;
}
/* Finally free the structure itself */
free((char *)ifp);
return 0;
}
/*
if_lookup - Given the ascii name of an interface, return a pointer to the
structure, or NULLIF if it doesn't exist
*/
struct iface *if_lookup(char *name)
{
struct iface *ifp;
for (ifp = Ifaces; ifp != NULLIF; ifp = ifp->next)
if (strcmp(ifp->name, name) == 0)
break;
return ifp;
}
/*
ismyaddr - Return iface pointer if 'addr' belongs to one of our interfaces,
NULLIF otherwise. This is used to tell if an incoming IP
datagram is for us, or if it has to be routed.
*/
struct iface *ismyaddr(int32 addr)
{
struct iface *ifp;
for (ifp = Ifaces; ifp != NULLIF; ifp = ifp->next)
if(addr == ifp->addr)
break;
return ifp;
}
/*
mask2width - Given a network mask, return the number of contiguous
1-bits starting from the most significant bit.
*/
static int mask2width(int32 mask)
{
int width,i;
width = 0;
for(i = 31; i >= 0; i--) {
if(!(mask & (1L << i)))
break;
width++;
}
return width;
}
/*
if_name - return buffer with name + comment
*/
char *if_name(struct iface *ifp, char *comment)
{
char *result = mallocw(strlen(ifp->name) + strlen(comment) + 1 );
strcpy(result, ifp->name );
return strcat(result, comment);
}
/* Raw output routine that tosses all packets. Used by dialer, tip, etc */
int bitbucket(struct iface *ifp, struct mbuf *bp)
{
free_p(bp);
return 0;
}